From 9738db88f68f251b9abc68b597f54e9c80fdec66 Mon Sep 17 00:00:00 2001 From: James Mckenzie Date: Thu, 14 Jul 2011 09:28:23 +0100 Subject: [PATCH] xen: Automatically find serial port on PCI/PCIe and AMT devices. Instead of having to manually look the right I/O port on the PCI devices or AMT devices, lets probe the card and find that automatically. This means that you don't have to have this: com1=115200,8n1,0xd800,0 But instead can have com1=115200,8n1,magic Or if you have AMT: com1=19200,8n1,amt Signed-off-by: James Mckenzie Signed-off-by: Konrad Rzeszutek Wilk Signed-off-by: Tom Goetz --- xen/drivers/char/ns16550.c | 73 +++++++++++++++++++++++++++++++++++++- 1 file changed, 72 insertions(+), 1 deletion(-) diff --git a/xen/drivers/char/ns16550.c b/xen/drivers/char/ns16550.c index c360ee70f5..db39f246c1 100644 --- a/xen/drivers/char/ns16550.c +++ b/xen/drivers/char/ns16550.c @@ -17,6 +17,8 @@ #include #include #include +#include +#include #include /* @@ -440,6 +442,64 @@ static int __init check_existence(struct ns16550 *uart) return (status == 0x90); } +static int +magic_uart_config (struct ns16550 *uart,int skip_amt) +{ + uint16_t class; + uint32_t bar0, len; + int b, d, f; + +/*Skanky hack - start at bus 1 to avoid AMT, a plug in card cannot be on bus 1 */ + + if (skip_amt) b=1; + else b=0; + + for (; b < 0x100; ++b) + { + for (d = 0; d < 0x20; ++d) + { + for (f = 0; f < 0x8; ++f) + { + + class = pci_conf_read16 (b, d, f, PCI_CLASS_DEVICE); + if (class != 0x700) + continue;; + + bar0 = pci_conf_read32 (b, d, f, PCI_BASE_ADDRESS_0); + + /* Not IO */ + if (!(bar0 & 1)) + continue; + + pci_conf_write32 (b, d, f, PCI_BASE_ADDRESS_0, 0xffffffff); + len = pci_conf_read32 (b, d, f, PCI_BASE_ADDRESS_0); + pci_conf_write32 (b, d, f, PCI_BASE_ADDRESS_0, bar0); + + /* Not 8 bytes */ + if ((len & 0xffff) != 0xfff9) + continue; + + uart->io_base = bar0 & 0xfffe; + uart->irq = 0; + + return 0; + + } + + } + } + + if (!skip_amt) + return -1; + + uart->io_base = 0x3f8; + uart->irq = 0; + uart->clock_hz = UART_CLOCK_HZ; + + return 0; +} + + #define PARSE_ERR(_f, _a...) \ do { \ printk( "ERROR: " _f "\n" , ## _a ); \ @@ -488,7 +548,18 @@ static void __init ns16550_parse_port_config( if ( *conf == ',' ) { conf++; - uart->io_base = simple_strtoul(conf, &conf, 0); + + if ( strncmp(conf,"magic",5) == 0 ) { + if (magic_uart_config(uart,1)) + return; + conf+=5; + } else if ( strncmp(conf,"amt",3) == 0 ) { + if (magic_uart_config(uart,0)) + return; + conf+=3; + } else { + uart->io_base = simple_strtoul(conf, &conf, 0); + } if ( *conf == ',' ) { -- 2.30.2